home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / MAIL.XPI / bin / chrome / messenger.jar / content / messenger / msgHdrViewOverlay.js < prev    next >
Encoding:
JavaScript  |  2005-12-22  |  50.9 KB  |  1,420 lines

  1. /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  2.  * ***** BEGIN LICENSE BLOCK *****
  3.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  4.  *
  5.  * The contents of this file are subject to the Mozilla Public License Version
  6.  * 1.1 (the "License"); you may not use this file except in compliance with
  7.  * the License. You may obtain a copy of the License at
  8.  * http://www.mozilla.org/MPL/
  9.  *
  10.  * Software distributed under the License is distributed on an "AS IS" basis,
  11.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  12.  * for the specific language governing rights and limitations under the
  13.  * License.
  14.  *
  15.  * The Original Code is Mozilla Communicator client code, released
  16.  * March 31, 1998.
  17.  *
  18.  * The Initial Developer of the Original Code is
  19.  * Netscape Communications Corporation.
  20.  * Portions created by the Initial Developer are Copyright (C) 1998-1999
  21.  * the Initial Developer. All Rights Reserved.
  22.  *
  23.  * Contributor(s):
  24.  *
  25.  * Alternatively, the contents of this file may be used under the terms of
  26.  * either of the GNU General Public License Version 2 or later (the "GPL"),
  27.  * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  28.  * in which case the provisions of the GPL or the LGPL are applicable instead
  29.  * of those above. If you wish to allow use of your version of this file only
  30.  * under the terms of either the GPL or the LGPL, and not to allow others to
  31.  * use your version of this file under the terms of the MPL, indicate your
  32.  * decision by deleting the provisions above and replace them with the notice
  33.  * and other provisions required by the GPL or the LGPL. If you do not delete
  34.  * the provisions above, a recipient may use your version of this file under
  35.  * the terms of any one of the MPL, the GPL or the LGPL.
  36.  *
  37.  * ***** END LICENSE BLOCK ***** */
  38.  
  39. /* This is where functions related to displaying the headers for a selected message in the
  40.    message pane live. */
  41.  
  42. ////////////////////////////////////////////////////////////////////////////////////
  43. // Warning: if you go to modify any of these JS routines please get a code review from
  44. // mscott@netscape.com. It's critical that the code in here for displaying
  45. // the message headers for a selected message remain as fast as possible. In particular, 
  46. // right now, we only introduce one reflow per message. i.e. if you click on a message in the thread
  47. // pane, we batch up all the changes for displaying the header pane (to, cc, attachements button, etc.) 
  48. // and we make a single pass to display them. It's critical that we maintain this one reflow per message
  49. // view in the message header pane. 
  50. ////////////////////////////////////////////////////////////////////////////////////
  51.  
  52. var msgHeaderParserContractID           = "@mozilla.org/messenger/headerparser;1";
  53. var abAddressCollectorContractID     = "@mozilla.org/addressbook/services/addressCollecter;1";
  54.  
  55. var gViewAllHeaders = false;
  56. var gNumAddressesToShow = 3;
  57. var gShowOrganization = false;
  58. var gShowUserAgent = false;
  59. var gCollectIncoming = false;
  60. var gCollectOutgoing = false;
  61. var gCollectNewsgroup = false;
  62. var gCollapsedHeaderViewMode = false;
  63. var gCollectAddressTimer = null;
  64. var gCollectAddress = null;
  65. var gBuildAttachmentsForCurrentMsg = false;
  66. var gBuildAttachmentPopupForCurrentMsg = true;
  67. var gBuiltExpandedView = false;
  68. var gBuiltCollapsedView = false;
  69. var gOpenLabel;
  70. var gOpenLabelAccesskey;
  71. var gSaveLabel;
  72. var gSaveLabelAccesskey;
  73. var gDetachLabel;
  74. var gDetachLabelAccesskey;
  75. var gDeleteLabel;
  76. var gDeleteLabelAccesskey;
  77. var gMessengerBundle;
  78. var gProfileDirURL;
  79. var gIOService;
  80. var gFileHandler;
  81.  
  82. var msgHeaderParser = Components.classes[msgHeaderParserContractID].getService(Components.interfaces.nsIMsgHeaderParser);
  83. var abAddressCollector = null;
  84.  
  85. // other components may listen to on start header & on end header notifications for each message we display
  86. // to do that you need to add yourself to our gMessageListeners array with object that has two properties:
  87. // onStartHeaders and onEndHeaders.
  88. var gMessageListeners = new Array;
  89.  
  90. // For every possible "view" in the message pane, you need to define the header names you want to
  91. // see in that view. In addition, include information describing how you want that header field to be
  92. // presented. i.e. if it's an email address field, if you want a toggle inserted on the node in case
  93. // of multiple email addresses, etc. We'll then use this static table to dynamically generate header view entries
  94. // which manipulate the UI. 
  95. // When you add a header to one of these view lists you can specify the following properties:
  96. // name: the name of the header. i.e. "to", "subject". This must be in lower case and the name of the
  97. //       header is used to help dynamically generate ids for objects in the document. (REQUIRED)
  98. // useToggle:      true if the values for this header are multiple email addresses and you want a 
  99. //                 a toggle icon to show a short vs. long list (DEFAULT: false)
  100. // useShortView:   (only works on some fields like From). If the field has a long presentation and a
  101. //                 short presentation we'll use the short one. i.e. if you are showing the From field and you
  102. //                 set this to true, we can show just "John Doe" instead of "John Doe <jdoe@netscape.net>".
  103. //                 (DEFAULT: false)
  104. // 
  105. // outputFunction: this is a method which takes a headerEntry (see the definition below) and a header value
  106. //                 This allows you to provide your own methods for actually determining how the header value
  107. //                 is displayed. (DEFAULT: updateHeaderValue which just sets the header value on the text node)
  108.  
  109. // Our first view is the collapsed view. This is very light weight view of the data. We only show a couple
  110. // fields.
  111. var gCollapsedHeaderList = [ {name:"subject", outputFunction:updateHeaderValueInTextNode},
  112.                              {name:"from", useShortView:true, outputFunction:OutputEmailAddresses},
  113.                              {name:"date", outputFunction:updateHeaderValueInTextNode}];
  114.  
  115. // We also have an expanded header view. This shows many of your more common (and useful) headers.
  116. var gExpandedHeaderList = [ {name:"subject"}, 
  117.                             {name:"from", outputFunction:OutputEmailAddresses},
  118.                             {name:"reply-to", outputFunction:OutputEmailAddresses},
  119.                             {name:"date"},
  120.                             {name:"to", useToggle:true, outputFunction:OutputEmailAddresses},
  121.                             {name:"cc", useToggle:true, outputFunction:OutputEmailAddresses},
  122.                             {name:"bcc", useToggle:true, outputFunction:OutputEmailAddresses},
  123.                             {name:"newsgroups", outputFunction:OutputNewsgroups},
  124.                             {name:"followup-to", outputFunction:OutputNewsgroups} ];
  125.  
  126. // Now, for each view the message pane can generate, we need a global table of headerEntries. These
  127. // header entry objects are generated dynamically based on the static date in the header lists (see above)
  128. // and elements we find in the DOM based on properties in the header lists. 
  129. var gCollapsedHeaderView = {};
  130. var gExpandedHeaderView  = {};
  131.  
  132. // currentHeaderData --> this is an array of header name and value pairs for the currently displayed message.
  133. //                       it's purely a data object and has no view information. View information is contained in the view objects.
  134. //                       for a given entry in this array you can ask for:
  135. // .headerName ---> name of the header (i.e. 'to'). Always stored in lower case
  136. // .headerValue --> value of the header "johndoe@netscape.net"
  137. var currentHeaderData = {};
  138.  
  139. // For the currently displayed message, we store all the attachment data. When displaying a particular
  140. // view, it's up to the view layer to extract this attachment data and turn it into something useful.
  141. // For a given entry in the attachments list, you can ask for the following properties:
  142. // .contentType --> the content type of the attachment
  143. // url --> an imap, or mailbox url which can be used to fetch the message
  144. // uri --> an RDF URI which refers to the message containig the attachment
  145. // isExternalAttachment --> boolean flag stating whether the attachment is external or not.
  146. var currentAttachments = new Array();
  147.  
  148. // createHeaderEntry --> our constructor method which creates a header Entry 
  149. // based on an entry in one of the header lists. A header entry is different from a header list.
  150. // a header list just describes how you want a particular header to be presented. The header entry
  151. // actually has knowledge about the DOM and the actual DOM elements associated with the header.
  152. // prefix --> the name of the view (i.e. "collapsed", "expanded")
  153. // headerListInfo --> entry from a header list.
  154. function createHeaderEntry(prefix, headerListInfo)
  155. {
  156.   var partialIDName = prefix + headerListInfo.name;
  157.   this.enclosingBox = document.getElementById(partialIDName + 'Box');
  158.   this.textNode = document.getElementById(partialIDName + 'Value');
  159.   this.isValid = false;
  160.  
  161.   if ("useToggle" in headerListInfo)
  162.   {
  163.     this.useToggle = headerListInfo.useToggle;
  164.     if (this.useToggle) // find the toggle icon in the document
  165.     {
  166.       this.toggleIcon = this.enclosingBox.toggleIcon;
  167.       this.longTextNode = this.enclosingBox.longEmailAddresses;
  168.       this.textNode = this.enclosingBox.emailAddresses;
  169.     }
  170.   }
  171.   else
  172.    this.useToggle = false;
  173.  
  174.   if ("useShortView" in headerListInfo)
  175.   {
  176.     this.useShortView = headerListInfo.useShortView;
  177.     this.enclosingBox.emailAddressNode = this.textNode;
  178.   }
  179.   else
  180.     this.useShortView = false;
  181.  
  182.   if ("outputFunction" in headerListInfo)
  183.     this.outputFunction = headerListInfo.outputFunction;
  184.   else
  185.     this.outputFunction = updateHeaderValue;
  186. }
  187.  
  188. function initializeHeaderViewTables()
  189. {
  190.   // iterate over each header in our header list arrays and create header entries 
  191.   // for each one. These header entries are then stored in the appropriate header table
  192.   var index;
  193.   for (index = 0; index < gCollapsedHeaderList.length; index++)
  194.   {
  195.     gCollapsedHeaderView[gCollapsedHeaderList[index].name] = 
  196.       new createHeaderEntry('collapsed', gCollapsedHeaderList[index]);
  197.   }
  198.  
  199.   for (index = 0; index < gExpandedHeaderList.length; index++)
  200.   {
  201.     var headerName = gExpandedHeaderList[index].name;
  202.     gExpandedHeaderView[headerName] = new createHeaderEntry('expanded', gExpandedHeaderList[index]);
  203.   }
  204.   
  205.   if (gShowOrganization)
  206.   {
  207.     var organizationEntry = {name:"organization", outputFunction:updateHeaderValue};
  208.     gExpandedHeaderView[organizationEntry.name] = new createHeaderEntry('expanded', organizationEntry);
  209.   }
  210.   
  211.   if (gShowUserAgent)
  212.   {
  213.     var userAgentEntry = {name:"user-agent", outputFunction:updateHeaderValue};
  214.     gExpandedHeaderView[userAgentEntry.name] = new createHeaderEntry('expanded', userAgentEntry);
  215.   }
  216. }
  217.  
  218. function OnLoadMsgHeaderPane()
  219. {
  220.   // HACK...force our XBL bindings file to be load before we try to create our first xbl widget....
  221.   // otherwise we have problems.
  222.  
  223.   document.loadBindingDocument('chrome://messenger/content/mailWidgets.xml');
  224.   
  225.   // load any preferences that at are global with regards to 
  226.   // displaying a message...
  227.   gNumAddressesToShow = pref.getIntPref("mailnews.max_header_display_length");
  228.   gCollectIncoming = pref.getBoolPref("mail.collect_email_address_incoming");
  229.   gCollectNewsgroup = pref.getBoolPref("mail.collect_email_address_newsgroup");
  230.   gCollectOutgoing = pref.getBoolPref("mail.collect_email_address_outgoing");
  231.   gShowUserAgent = pref.getBoolPref("mailnews.headers.showUserAgent");
  232.   gShowOrganization = pref.getBoolPref("mailnews.headers.showOrganization");
  233.   initializeHeaderViewTables();
  234.  
  235.   var toggleHeaderView = document.getElementById("msgHeaderView");
  236.   var initialCollapsedSetting = toggleHeaderView.getAttribute("state");
  237.   if (initialCollapsedSetting == "true")
  238.     gCollapsedHeaderViewMode = true;   
  239.  
  240.   // dispatch an event letting any listeners know that we have loaded the message pane
  241.   var event = document.createEvent('Events');
  242.   event.initEvent('messagepane-loaded', false, true);
  243.   var headerViewElement = document.getElementById("msgHeaderView");
  244.   headerViewElement.dispatchEvent(event);
  245. }
  246.  
  247. function OnUnloadMsgHeaderPane()
  248. {
  249.   // dispatch an event letting any listeners know that we have unloaded the message pane
  250.   var event = document.createEvent('Events');
  251.   event.initEvent('messagepane-unloaded', false, true);
  252.   var headerViewElement = document.getElementById("msgHeaderView");
  253.   headerViewElement.dispatchEvent(event);
  254. }
  255.  
  256. // The messageHeaderSink is the class that gets notified of a message's headers as we display the message
  257. // through our mime converter. 
  258.  
  259. var messageHeaderSink = {
  260.     onStartHeaders: function()
  261.     {
  262.       this.mSaveHdr = null;
  263.       // clear out any pending collected address timers...
  264.       if (gCollectAddressTimer)
  265.       {
  266.         gCollectAddress = "";        
  267.         clearTimeout(gCollectAddressTimer);
  268.         gCollectAddressTimer = null;
  269.       }
  270.  
  271.       // every time we start to redisplay a message, check the view all headers pref....
  272.       var showAllHeadersPref = pref.getIntPref("mail.show_headers");
  273.       if (showAllHeadersPref == 2)
  274.       {
  275.         gViewAllHeaders = true;
  276.       }
  277.       else
  278.       {
  279.         if (gViewAllHeaders) // if we currently are in view all header mode, rebuild our header view so we remove most of the header data
  280.         { 
  281.           hideHeaderView(gExpandedHeaderView);
  282.           gExpandedHeaderView = {};
  283.           initializeHeaderViewTables(); 
  284.         }
  285.                 
  286.         gViewAllHeaders = false;
  287.       }
  288.  
  289.       ClearCurrentHeaders();
  290.       gBuiltExpandedView = false;
  291.       gBuiltCollapsedView = false;
  292.       gBuildAttachmentsForCurrentMsg = false;
  293.       gBuildAttachmentPopupForCurrentMsg = true;
  294.       ClearAttachmentList();
  295.       ClearEditMessageButton();
  296.       gMessageNotificationBar.clearMsgNotifications();
  297.  
  298.       for (var index in gMessageListeners)
  299.         gMessageListeners[index].onStartHeaders();
  300.     },
  301.  
  302.     onEndHeaders: function() 
  303.     {
  304.       // WARNING: This is the ONLY routine inside of the message Header Sink that should 
  305.       // trigger a reflow!
  306.       CheckNotify();
  307.       
  308.       ClearHeaderView(gCollapsedHeaderView);
  309.       ClearHeaderView(gExpandedHeaderView);
  310.  
  311.       EnsureSubjectValue(); // make sure there is a subject even if it's empty so we'll show the subject and the twisty
  312.       
  313.       ShowMessageHeaderPane();
  314.       UpdateMessageHeaders();
  315.       if (gIsEditableMsgFolder)
  316.         ShowEditMessageButton();
  317.  
  318.       for (var index in gMessageListeners)
  319.         gMessageListeners[index].onEndHeaders();
  320.     },
  321.  
  322.     processHeaders: function(headerNameEnumerator, headerValueEnumerator, dontCollectAddress)
  323.     {
  324.       this.onStartHeaders(); 
  325.  
  326.       var index = 0;
  327.       while (headerNameEnumerator.hasMore()) 
  328.       {
  329.         var header = new Object;        
  330.         header.headerValue = headerValueEnumerator.getNext();
  331.         header.headerName = headerNameEnumerator.getNext();
  332.  
  333.         // for consistancy sake, let's force all header names to be lower case so
  334.         // we don't have to worry about looking for: Cc and CC, etc.
  335.         var lowerCaseHeaderName = header.headerName.toLowerCase();
  336.  
  337.         // if we have an x-mailer, x-newsreader, or x-mimeole string,
  338.         // put it in the user-agent slot which we know how to handle already.
  339.         if (lowerCaseHeaderName == "x-mailer" ||
  340.             lowerCaseHeaderName == "x-newsreader" ||
  341.             lowerCaseHeaderName == "x-mimeole")
  342.           lowerCaseHeaderName = "user-agent";          
  343.         
  344.         if (this.mDummyMsgHeader)
  345.         {
  346.           if (lowerCaseHeaderName == "from")
  347.             this.mDummyMsgHeader.author = header.headerValue;
  348.           else if (lowerCaseHeaderName == "to")
  349.             this.mDummyMsgHeader.recipients = header.headerValue;
  350.           else if (lowerCaseHeaderName == "cc")
  351.             this.mDummyMsgHeader.ccList = header.headerValue;
  352.           else if (lowerCaseHeaderName == "subject")
  353.             this.mDummyMsgHeader.subject = header.headerValue;
  354.           else if (lowerCaseHeaderName == "reply-to")
  355.             this.mDummyMsgHeader.replyTo = header.headerValue;
  356.           else if (lowerCaseHeaderName == "message-id")
  357.             this.mDummyMsgHeader.messageId = header.headerValue;
  358.  
  359.         }
  360.         // according to RFC 2822, certain headers
  361.         // can occur "unlimited" times
  362.         if (lowerCaseHeaderName in currentHeaderData)
  363.         {
  364.           // sometimes, you can have multiple To or Cc lines....
  365.           // in this case, we want to append these headers into one.
  366.           if (lowerCaseHeaderName == 'to' || lowerCaseHeaderName == 'cc')
  367.             currentHeaderData[lowerCaseHeaderName].headerValue = currentHeaderData[lowerCaseHeaderName].headerValue + ',' + header.headerValue;
  368.           else 
  369.           {  
  370.             // use the index to create a unique header name like:
  371.             // received5, received6, etc
  372.             currentHeaderData[lowerCaseHeaderName + index++] = header;
  373.           }
  374.         }
  375.         else
  376.          currentHeaderData[lowerCaseHeaderName] = header;
  377.  
  378.         if (lowerCaseHeaderName == "from")
  379.         {
  380.           if (header.value) 
  381.           {
  382.             if ((gCollectIncoming && !dontCollectAddress) || 
  383.                 (gCollectNewsgroup && dontCollectAddress))
  384.             {
  385.               if (!abAddressCollector)
  386.                 abAddressCollector = Components.classes[abAddressCollectorContractID].getService(Components.interfaces.nsIAbAddressCollecter);
  387.  
  388.               gCollectAddress = header.headerValue;
  389.               // collect, and add card if doesn't exist, unknown preferred send format
  390.               gCollectAddressTimer = setTimeout('abAddressCollector.collectUnicodeAddress(gCollectAddress, true, Components.interfaces.nsIAbPreferMailFormat.unknown);', 2000);
  391.             }
  392.             else if (gCollectOutgoing) 
  393.             {
  394.               if (!abAddressCollector)
  395.                 abAddressCollector = Components.classes[abAddressCollectorContractID].getService(Components.interfaces.nsIAbAddressCollecter);
  396.  
  397.               // collect, but only update existing cards, unknown preferred send format
  398.               gCollectAddress = header.headerValue;
  399.               gCollectAddressTimer = setTimeout('abAddressCollector.collectUnicodeAddress(gCollectAddress, false, Components.interfaces.nsIAbPreferMailFormat.unknown);', 2000);
  400.             }
  401.           }
  402.         } // if lowerCaseHeaderName == "from"
  403.       } // while we have more headers to parse
  404.  
  405.       this.onEndHeaders();
  406.     },
  407.  
  408.     handleAttachment: function(contentType, url, displayName, uri, isExternalAttachment) 
  409.     {
  410.       // presentation level change....don't show vcards as external attachments in the UI.
  411.       // libmime already renders them inline.
  412.  
  413.       try
  414.       {
  415.         if (!this.mSaveHdr)
  416.           this.mSaveHdr = messenger.msgHdrFromURI(uri);
  417.       }
  418.       catch (ex) {}
  419.       if (contentType == "text/x-vcard")
  420.       {
  421.         var inlineAttachments = pref.getBoolPref("mail.inline_attachments");
  422.         var displayHtmlAs = pref.getIntPref("mailnews.display.html_as");
  423.         if (inlineAttachments && !displayHtmlAs)
  424.           return;
  425.       }
  426.  
  427.       currentAttachments.push (new createNewAttachmentInfo(contentType, url, displayName, uri, isExternalAttachment));
  428.       // if we have an attachment, set the MSG_FLAG_ATTACH flag on the hdr
  429.       // this will cause the "message with attachment" icon to show up
  430.       // in the thread pane
  431.       // we only need to do this on the first attachment
  432.       var numAttachments = currentAttachments.length;
  433.       if (numAttachments == 1) {
  434.         // we also have to enable the File/Attachments menuitem
  435.         var node = document.getElementById("fileAttachmentMenu");
  436.         if (node)
  437.           node.removeAttribute("disabled");
  438.  
  439.         try {
  440.           // convert the uri into a hdr
  441.           this.mSaveHdr.markHasAttachments(true);
  442.         }
  443.         catch (ex) {
  444.           dump("ex = " + ex + "\n");
  445.         }
  446.       }
  447.     },
  448.     
  449.     onEndAllAttachments: function()
  450.     {
  451.       // AddSaveAllAttachmentsMenu();
  452.       if (gCollapsedHeaderViewMode)
  453.         displayAttachmentsForCollapsedView();
  454.       else
  455.         displayAttachmentsForExpandedView();
  456.     },
  457.  
  458.     onEndMsgDownload: function(url)
  459.     {
  460.       // if we don't have any attachments, turn off the attachments flag
  461.       if (!this.mSaveHdr)
  462.       {
  463.         var messageUrl = url.QueryInterface(Components.interfaces.nsIMsgMessageUrl);
  464.         try
  465.         {
  466.           this.mSaveHdr = messenger.msgHdrFromURI(messageUrl.uri);
  467.         }
  468.         catch (ex) {}
  469.  
  470.       }
  471.       if (!currentAttachments.length && this.mSaveHdr)
  472.         this.mSaveHdr.markHasAttachments(false);
  473.       OnMsgParsed(url);
  474.     },
  475.  
  476.     onEndMsgHeaders: function(url)
  477.     { 
  478.       OnMsgLoaded(url);
  479.     },
  480.     
  481.     onMsgHasRemoteContent: function(aMsgHdr)
  482.     {
  483.       gMessageNotificationBar.setRemoteContentMsg(aMsgHdr);
  484.     },
  485.  
  486.     mSecurityInfo  : null,
  487.     mSaveHdr: null,
  488.     getSecurityInfo: function()
  489.     {
  490.       return this.mSecurityInfo;
  491.     },
  492.     setSecurityInfo: function(aSecurityInfo)
  493.     {
  494.       this.mSecurityInfo = aSecurityInfo;
  495.     },
  496.  
  497.     mDummyMsgHeader: null,
  498.  
  499.     getDummyMsgHeader: function()
  500.     {
  501.       if (!this.mDummyMsgHeader)
  502.         this.mDummyMsgHeader = new nsDummyMsgHeader();
  503.       return this.mDummyMsgHeader;
  504.     }
  505. };
  506.  
  507. function EnsureSubjectValue()
  508. {
  509.   if (!('subject' in currentHeaderData))
  510.   {
  511.     var foo = new Object;
  512.     foo.headerValue = "";
  513.     foo.headerName = 'subject';
  514.     currentHeaderData[foo.headerName] = foo;
  515.   } 
  516. }
  517.  
  518. function CheckNotify()
  519. {
  520.   if ("NotifyClearAddresses" in this)
  521.     NotifyClearAddresses();
  522. }
  523.  
  524.  
  525. // flush out any local state being held by a header entry for a given
  526. // table
  527. function ClearHeaderView(headerTable)
  528. {
  529.   for (index in headerTable)
  530.   {
  531.      var headerEntry = headerTable[index];
  532.      if (headerEntry.useToggle)
  533.      {
  534.        headerEntry.enclosingBox.clearEmailAddresses();    
  535.      }
  536.  
  537.      headerEntry.valid = false;
  538.   }
  539. }
  540.  
  541. // make sure that any valid header entry in the table is collapsed
  542. function hideHeaderView(headerTable)
  543. {
  544.   for (index in headerTable)
  545.   {
  546.     headerTable[index].enclosingBox.collapsed = true;
  547.   }
  548. }
  549.  
  550. // make sure that any valid header entry in the table specified is 
  551. // visible
  552. function showHeaderView(headerTable)
  553. {
  554.   var headerEntry;
  555.   for (index in headerTable)
  556.   {
  557.     headerEntry = headerTable[index];
  558.     if (headerEntry.valid)
  559.     {
  560.       headerEntry.enclosingBox.collapsed = false;
  561.     }
  562.     else // if the entry is invalid, always make sure it's collapsed
  563.       headerEntry.enclosingBox.collapsed = true;
  564.   }
  565. }
  566.  
  567. // make sure the appropriate fields within the currently displayed view header mode
  568. // are collapsed or visible...
  569. function updateHeaderViews()
  570. {
  571.   if (gCollapsedHeaderViewMode)
  572.   {
  573.     showHeaderView(gCollapsedHeaderView);
  574.     displayAttachmentsForCollapsedView();
  575.   }
  576.   else
  577.   {
  578.     showHeaderView(gExpandedHeaderView);
  579.     displayAttachmentsForExpandedView();
  580.   }
  581. }
  582.  
  583. function ToggleHeaderView ()
  584. {
  585.   var expandedNode = document.getElementById("expandedHeaderView");
  586.   var collapsedNode = document.getElementById("collapsedHeaderView");
  587.   var toggleHeaderView = document.getElementById("msgHeaderView");
  588.  
  589.   if (gCollapsedHeaderViewMode)
  590.   {          
  591.     gCollapsedHeaderViewMode = false;
  592.     // hide the current view
  593.     hideHeaderView(gCollapsedHeaderView);
  594.     // update the current view
  595.     UpdateMessageHeaders();
  596.     
  597.     // now uncollapse / collapse the right views
  598.     expandedNode.collapsed = false;
  599.     collapsedNode.collapsed = true;
  600.   }
  601.   else
  602.   {
  603.     gCollapsedHeaderViewMode = true;
  604.     // hide the current view
  605.     hideHeaderView(gExpandedHeaderView);
  606.     // update the current view
  607.     UpdateMessageHeaders();
  608.     
  609.     // now uncollapse / collapse the right views
  610.     collapsedNode.collapsed = false;
  611.     expandedNode.collapsed = true;
  612.   }  
  613.  
  614.   if (gCollapsedHeaderViewMode)
  615.     toggleHeaderView.setAttribute("state", "true");
  616.   else
  617.     toggleHeaderView.setAttribute("state", "false");
  618. }
  619.  
  620. // default method for updating a header value into a header entry
  621. function updateHeaderValue(headerEntry, headerValue)
  622. {
  623.   headerEntry.enclosingBox.headerValue = headerValue;
  624. }
  625.  
  626. function updateHeaderValueInTextNode(headerEntry, headerValue)
  627. {
  628.   headerEntry.textNode.value = headerValue;
  629. }
  630.  
  631. function createNewHeaderView(headerName)
  632. {
  633.   var idName = 'expanded' + headerName + 'Box';
  634.   var newHeader = document.createElement("mail-headerfield");
  635.   newHeader.setAttribute('id', idName);
  636.   newHeader.setAttribute('label', currentHeaderData[headerName].headerName + ':');
  637.   // all mail-headerfield elements are keyword related
  638.   newHeader.setAttribute('keywordrelated','true');
  639.   newHeader.collapsed = true;
  640.  
  641.   // this new element needs to be inserted into the view...
  642.   var topViewNode = document.getElementById('expandedHeaders');
  643.  
  644.   topViewNode.appendChild(newHeader);
  645.   
  646.   this.enclosingBox = newHeader
  647.   this.isValid = false;
  648.   this.useToggle = false;
  649.   this.useShortView = false;
  650.   this.outputFunction = updateHeaderValue;
  651. }
  652.  
  653. // UpdateMessageHeaders: Iterate through all the current header data we received from mime for this message
  654. // for each header entry table, see if we have a corresponding entry for that header. i.e. does the particular
  655. // view care about this header value. if it does then call updateHeaderEntry
  656. function UpdateMessageHeaders()
  657. {
  658.   // iterate over each header we received and see if we have a matching entry in each
  659.   // header view table...
  660.  
  661.   for (headerName in currentHeaderData)
  662.   {
  663.     var headerField = currentHeaderData[headerName];
  664.     var headerEntry = null;
  665.  
  666.     if (headerName == "subject")
  667.     {
  668.       try {
  669.         if (gDBView.keyForFirstSelectedMessage == nsMsgKey_None)
  670.         {
  671.           var folder = null;
  672.           if (gCurrentFolderUri)
  673.             folder = RDF.GetResource(gCurrentFolderUri).QueryInterface(Components.interfaces.nsIMsgFolder);
  674.           setTitleFromFolder(folder, headerField.headerValue);
  675.         }
  676.       } catch (ex) {}
  677.     }
  678.     
  679.     if (gCollapsedHeaderViewMode && !gBuiltCollapsedView)
  680.     { 
  681.       if (headerName in gCollapsedHeaderView)
  682.         headerEntry = gCollapsedHeaderView[headerName];
  683.     }
  684.     else if (!gCollapsedHeaderViewMode && !gBuiltExpandedView)
  685.     {
  686.       if (headerName in gExpandedHeaderView)
  687.         headerEntry = gExpandedHeaderView[headerName];
  688.  
  689.       if (!headerEntry && gViewAllHeaders)
  690.       {
  691.         // for view all headers, if we don't have a header field for this value....cheat and create one....then
  692.         // fill in a headerEntry
  693.         gExpandedHeaderView[headerName] = new createNewHeaderView(headerName);
  694.         headerEntry = gExpandedHeaderView[headerName];
  695.       }
  696.     } // if we are in expanded view....
  697.  
  698.     if (headerEntry)
  699.     {  
  700.       headerEntry.outputFunction(headerEntry, headerField.headerValue);
  701.       headerEntry.valid = true;    
  702.     }
  703.   }
  704.  
  705.   if (gCollapsedHeaderViewMode)
  706.    gBuiltCollapsedView = true;
  707.   else
  708.    gBuiltExpandedView = true;
  709.  
  710.   // now update the view to make sure the right elements are visible
  711.   updateHeaderViews();
  712.   
  713.   if ("FinishEmailProcessing" in this)
  714.     FinishEmailProcessing();
  715. }
  716.  
  717. function ClearCurrentHeaders()
  718. {
  719.   currentHeaderData = {};
  720.   currentAttachments = new Array();
  721. }
  722.  
  723. function ShowMessageHeaderPane()
  724.   var node;
  725.   if (gCollapsedHeaderViewMode)
  726.   {          
  727.     node = document.getElementById("collapsedHeaderView");
  728.     if (node)
  729.       node.collapsed = false;
  730.   }
  731.   else
  732.   {
  733.     node = document.getElementById("expandedHeaderView");
  734.     if (node)
  735.       node.collapsed = false;
  736.   }
  737.  
  738.     /* workaround for 39655 */
  739.   if (gFolderJustSwitched) 
  740.   {
  741.     var el = document.getElementById("msgHeaderView");
  742.     el.setAttribute("style", el.getAttribute("style"));
  743.     gFolderJustSwitched = false;    
  744.   }
  745. }
  746.  
  747. function HideMessageHeaderPane()
  748. {
  749.   var node = document.getElementById("collapsedHeaderView");
  750.   if (node)
  751.     node.collapsed = true;
  752.  
  753.   node = document.getElementById("expandedHeaderView");
  754.   if (node)
  755.     node.collapsed = true;
  756.  
  757.   // we also have to disable the File/Attachments menuitem
  758.   node = document.getElementById("fileAttachmentMenu");
  759.   if (node)
  760.     node.setAttribute("disabled", "true");
  761. }
  762.  
  763. function OutputNewsgroups(headerEntry, headerValue)
  764.   headerValue = headerValue.replace(/,/g,", ");
  765.   updateHeaderValue(headerEntry, headerValue);
  766. }
  767.  
  768. // OutputEmailAddresses --> knows how to take a comma separated list of email addresses,
  769. // extracts them one by one, linkifying each email address into a mailto url. 
  770. // Then we add the link'ified email address to the parentDiv passed in.
  771. // 
  772. // defaultParentDiv --> the div to add the link-ified email addresses into. 
  773. // emailAddresses --> comma separated list of the addresses for this header field
  774. // includeShortLongToggle --> true if you want to include the ability to toggle between short/long
  775. // address views for this header field. If true, then pass in a another div which is the div the long
  776. // view will be added too...
  777. // useShortView --> if true, we'll only generate the Name of the email address field instead of
  778. //                        showing the name + the email address.
  779.  
  780. function OutputEmailAddresses(headerEntry, emailAddresses)
  781. {
  782.   if ( !emailAddresses ) return;
  783.  
  784.   if (msgHeaderParser)
  785.   {
  786.     var addresses = {};
  787.     var fullNames = {};
  788.     var names = {};
  789.     var numAddresses =  0;
  790.  
  791.     numAddresses = msgHeaderParser.parseHeadersWithArray(emailAddresses, addresses, names, fullNames);
  792.     var index = 0;
  793.     while (index < numAddresses)
  794.     {
  795.       // if we want to include short/long toggle views and we have a long view, always add it.
  796.       // if we aren't including a short/long view OR if we are and we haven't parsed enough
  797.       // addresses to reach the cutoff valve yet then add it to the default (short) div.
  798.       if (headerEntry.useToggle)
  799.       {
  800.         var address = {};
  801.         address.emailAddress = addresses.value[index];
  802.         address.fullAddress = fullNames.value[index];
  803.         address.displayName = names.value[index];
  804.         headerEntry.enclosingBox.addAddressView(address);
  805.       }
  806.       else
  807.       {
  808.         updateEmailAddressNode(headerEntry.enclosingBox.emailAddressNode, addresses.value[index], 
  809.                                fullNames.value[index], names.value[index], headerEntry.useShortView);
  810.       }
  811.  
  812.       if (headerEntry.enclosingBox.getAttribute("id") == "expandedfromBox") {
  813.         setFromBuddyIcon(addresses.value[index]);
  814.       }
  815.  
  816.       index++;
  817.     }
  818.     
  819.     if (headerEntry.useToggle)
  820.       headerEntry.enclosingBox.buildViews(gNumAddressesToShow);
  821.   } // if msgheader parser
  822. }
  823.  
  824. function setFromBuddyIcon(email)
  825. {
  826.    var fromBuddyIcon = document.getElementById("fromBuddyIcon");
  827.  
  828.    try {
  829.      // better to cache this?
  830.      var myScreenName = pref.getCharPref("aim.session.screenname");
  831.  
  832.      if (!abAddressCollector)
  833.        abAddressCollector = Components.classes[abAddressCollectorContractID].getService(Components.interfaces.nsIAbAddressCollecter);
  834.  
  835.      var card = abAddressCollector.getCardFromAttribute("PrimaryEmail", email);
  836.  
  837.      if (myScreenName && card && card.aimScreenName) {
  838.        if (!gIOService) {
  839.          // lazily create these globals
  840.          gIOService = Components.classes["@mozilla.org/network/io-service;1"].getService(Components.interfaces.nsIIOService);
  841.          gFileHandler = gIOService.getProtocolHandler("file").QueryInterface(Components.interfaces.nsIFileProtocolHandler);
  842.          
  843.          var dirService = Components.classes["@mozilla.org/file/directory_service;1"]
  844.              .getService(Components.interfaces.nsIProperties);
  845.          var profileDir = dirService.get("ProfD", Components.interfaces.nsIFile);
  846.          gProfileDirURL = gIOService.newFileURI(profileDir);
  847.        }
  848.  
  849.        // if we did have a buddy icon on disk for this screenname, this would be the file url spec for it
  850.        var iconURLStr = gProfileDirURL.spec + "/NIM/" + myScreenName + "/picture/" + card.aimScreenName + ".gif";
  851.  
  852.        // check if the file exists
  853.        // is this a perf hit?  (how expensive is stat()?)
  854.        var iconFile = gFileHandler.getFileFromURLSpec(iconURLStr);
  855.        if (iconFile.exists()) {
  856.          fromBuddyIcon.setAttribute("src", iconURLStr);
  857.          return;
  858.        }
  859.      }
  860.    }
  861.    catch (ex) {
  862.      // can get here if no screenname
  863.      //dump("ex = " + ex + "\n");
  864.    }
  865.    fromBuddyIcon.setAttribute("src", "");
  866. }
  867.  
  868. function updateEmailAddressNode(emailAddressNode, emailAddress, fullAddress, displayName, useShortView)
  869. {
  870.   if (useShortView && displayName) {
  871.     emailAddressNode.setAttribute("label", displayName);
  872.     emailAddressNode.setAttribute("tooltiptext", emailAddress);
  873.   } else {
  874.     emailAddressNode.setAttribute("label", fullAddress || displayName);
  875.     emailAddressNode.removeAttribute("tooltiptext");
  876.   }
  877.   emailAddressNode.setTextAttribute("emailAddress", emailAddress);
  878.   emailAddressNode.setTextAttribute("fullAddress", fullAddress);  
  879.   emailAddressNode.setTextAttribute("displayName", displayName);  
  880.   
  881.   if ("AddExtraAddressProcessing" in this)
  882.     AddExtraAddressProcessing(emailAddress, emailAddressNode);
  883. }
  884.  
  885. // createnewAttachmentInfo --> constructor method for creating new attachment object which goes into the
  886. // data attachment array.
  887. function createNewAttachmentInfo(contentType, url, displayName, uri, isExternalAttachment)
  888. {
  889.   this.contentType = contentType;
  890.   this.url = url;
  891.   this.displayName = displayName;
  892.   this.uri = uri;
  893.   this.isExternalAttachment = isExternalAttachment;
  894.   this.attachment = this;
  895. }
  896.  
  897. createNewAttachmentInfo.prototype.saveAttachment = function saveAttachment()
  898. {
  899.   if (this.isExternalAttachment)
  900.     internalSave(this.url, null,
  901.                  this.displayName, null,
  902.                  this.contentType, false,
  903.                  "SaveAttachmentTitle", null, null);
  904.   else
  905.     messenger.saveAttachment(this.contentType, 
  906.                              this.url, 
  907.                              encodeURIComponent(this.displayName), 
  908.                              this.uri,
  909.                              false);
  910. }
  911.  
  912. createNewAttachmentInfo.prototype.openAttachment = function openAttachment()
  913. {
  914.   messenger.openAttachment(this.contentType, 
  915.                            this.url, 
  916.                            encodeURIComponent(this.displayName), 
  917.                            this.uri,
  918.                            this.isExternalAttachment);
  919. }
  920.  
  921. createNewAttachmentInfo.prototype.printAttachment = function printAttachment()
  922. {
  923.   /* we haven't implemented the ability to print attachments yet...
  924.   messenger.printAttachment(this.contentType, 
  925.                             this.url, 
  926.                             encodeURIComponent(this.displayName), 
  927.                             this.uri);
  928.   */
  929. }
  930.  
  931. createNewAttachmentInfo.prototype.deleteAttachment = function deleteAttachment()
  932. {
  933.   messenger.detachAttachment(this.contentType,
  934.                              this.url,
  935.                              encodeURIComponent(this.displayName),
  936.                              this.uri,
  937.                              false);
  938. }
  939.  
  940. createNewAttachmentInfo.prototype.detachAttachment = function detachAttachment()
  941. {
  942.   messenger.detachAttachment(this.contentType,
  943.                              this.url,
  944.                              encodeURIComponent(this.displayName),
  945.                              this.uri,
  946.                              true);
  947. }
  948.  
  949. function CanDetachAttachments()
  950. {
  951.   var uri = GetLoadedMessage();
  952.   return !IsNewsMessage(uri) && (!IsImapMessage(uri) || CheckOnline());
  953. }
  954.  
  955. function onShowAttachmentContextMenu()
  956. {
  957.   // if no attachments are selected, disable the Open and Save...
  958.   var attachmentList = document.getElementById('attachmentList');
  959.   var selectedAttachments = attachmentList.selectedItems;
  960.   var openMenu = document.getElementById('context-openAttachment');
  961.   var saveMenu = document.getElementById('context-saveAttachment');
  962.   var detachMenu = document.getElementById('context-detachAttachment');
  963.   var deleteMenu = document.getElementById('context-deleteAttachment');
  964.   var detachAllMenu = document.getElementById('context-detachAllAttachments');
  965.   var deleteAllMenu = document.getElementById('context-deleteAllAttachments');
  966.   var canDetach = CanDetachAttachments();
  967.   var canOpen = false;
  968.   for (var i = 0; i < selectedAttachments.length && !canOpen; i++)
  969.     canOpen = selectedAttachments[i].attachment.contentType != 'text/x-moz-deleted';
  970.   if (canOpen && selectedAttachments.length == 1)
  971.   {
  972.     openMenu.removeAttribute('disabled');
  973.   }
  974.   else
  975.   {
  976.     openMenu.setAttribute('disabled', true);
  977.   }
  978.   if (canOpen)
  979.   {
  980.     saveMenu.removeAttribute('disabled');
  981.   }
  982.   else
  983.   {
  984.     saveMenu.setAttribute('disabled', true);
  985.   }
  986.   if (canDetach && canOpen)
  987.   {
  988.     detachMenu.removeAttribute('disabled');
  989.     deleteMenu.removeAttribute('disabled');
  990.   }
  991.   else
  992.   {
  993.     detachMenu.setAttribute('disabled', 'true');
  994.     deleteMenu.setAttribute('disabled', 'true');
  995.   }
  996.   if (canDetach)
  997.   {
  998.     detachAllMenu.removeAttribute('disabled');
  999.     deleteAllMenu.removeAttribute('disabled');
  1000.   }
  1001.   else
  1002.   {
  1003.     detachAllMenu.setAttribute('disabled', 'true');
  1004.     deleteAllMenu.setAttribute('disabled', 'true');
  1005.   }
  1006. }
  1007.  
  1008. // this is our onclick handler for the attachment list. 
  1009. // A double click in a listitem simulates "opening" the attachment....
  1010. function attachmentListClick(event)
  1011.     // we only care about button 0 (left click) events
  1012.     if (event.button != 0) return;
  1013.  
  1014.     if (event.detail == 2) // double click
  1015.     {
  1016.       var target = event.target;
  1017.       if (target.localName == "listitem")
  1018.       {
  1019.     target.attachment.openAttachment();
  1020.       }
  1021.     }
  1022. }
  1023.  
  1024. // on command handlers for the attachment list context menu...
  1025. // commandPrefix matches one of our existing functions 
  1026. // (openAttachment, saveAttachment, etc.)
  1027. function handleAttachmentSelection(commandPrefix)
  1028. {
  1029.   var attachmentList = document.getElementById('attachmentList');
  1030.   var selectedAttachments = attachmentList.selectedItems;
  1031.   if (selectedAttachments.length > 1)
  1032.     HandleMultipleAttachments(commandPrefix, selectedAttachments);
  1033.   else
  1034.     selectedAttachments[0].attachment[commandPrefix]();
  1035. }
  1036.  
  1037. function displayAttachmentsForExpandedView()
  1038. {
  1039.   var numAttachments = currentAttachments.length;
  1040.   if (numAttachments > 0 && !gBuildAttachmentsForCurrentMsg)
  1041.   {
  1042.     var attachmentList = document.getElementById('attachmentList');
  1043.  
  1044.     for (index in currentAttachments)
  1045.     {
  1046.       var attachment = currentAttachments[index];
  1047.  
  1048.       // we need to create a listitem to insert the attachment
  1049.       // into the attachment list..
  1050.       var item = attachmentList.appendItem(attachment.displayName,"");
  1051.       item.setAttribute("class", "listitem-iconic attachment-item"); 
  1052.       item.setAttribute("tooltip", "attachmentListTooltip");
  1053.       item.attachment = attachment;
  1054.       item.setAttribute("attachmentUrl", attachment.url);
  1055.       item.setAttribute("attachmentContentType", attachment.contentType);
  1056.       item.setAttribute("attachmentUri", attachment.uri);
  1057.       if (attachment.contentType == "text/x-moz-deleted")
  1058.         item.setAttribute('disabled', 'true');
  1059.       else
  1060.         setApplicationIconForAttachment(attachment, item);
  1061.     } // for each attachment   
  1062.  
  1063.     gBuildAttachmentsForCurrentMsg = true;
  1064.   }
  1065.    
  1066.   var expandedAttachmentBox = document.getElementById('expandedAttachmentBox');
  1067.   expandedAttachmentBox.collapsed = numAttachments <= 0;
  1068. }
  1069.  
  1070. // attachment --> the attachment struct containing all the information on the attachment
  1071. // listitem --> the listitem currently showing the attachment.
  1072. function setApplicationIconForAttachment(attachment, listitem)
  1073. {
  1074.    // generate a moz-icon url for the attachment so we'll show a nice icon next to it.
  1075.    listitem.setAttribute('image', "moz-icon:" + "//" + attachment.displayName + "?size=16&contentType=" + attachment.contentType);
  1076. }
  1077.  
  1078. function displayAttachmentsForCollapsedView()
  1079. {
  1080.   var numAttachments = currentAttachments.length;
  1081.   var attachmentNode = document.getElementById('collapsedAttachmentBox');
  1082.   attachmentNode.collapsed = numAttachments <= 0; // make sure the attachment button is visible
  1083. }
  1084.  
  1085. // Public method called to generate a tooltip over an attachment
  1086. function FillInAttachmentTooltip(cellNode)
  1087. {
  1088.   var attachmentName = cellNode.getAttribute("label");
  1089.   var tooltipNode = document.getElementById("attachmentListTooltip");
  1090.   tooltipNode.setAttribute("label", attachmentName);
  1091.   return true;
  1092. }
  1093.  
  1094. // Public method called when we create the attachments file menu
  1095. function FillAttachmentListPopup(popup)
  1096. {
  1097.   // the FE sometimes call this routine TWICE...I haven't been able to figure out why yet...
  1098.   // protect against it...
  1099.  
  1100.   if (!gBuildAttachmentPopupForCurrentMsg) return; 
  1101.  
  1102.   var attachmentIndex = 0;
  1103.  
  1104.   // otherwise we need to build the attachment view...
  1105.   // First clear out the old view...
  1106.   ClearAttachmentMenu(popup);
  1107.  
  1108.   for (index in currentAttachments)
  1109.   {
  1110.     ++attachmentIndex;
  1111.     addAttachmentToPopup(popup, currentAttachments[index], attachmentIndex);
  1112.   }
  1113.  
  1114.   gBuildAttachmentPopupForCurrentMsg = false;
  1115.  
  1116.   var detachAllMenu = document.getElementById('file-detachAllAttachments');
  1117.   var deleteAllMenu = document.getElementById('file-deleteAllAttachments');
  1118.   if (CanDetachAttachments())
  1119.   {
  1120.     detachAllMenu.removeAttribute('disabled');
  1121.     deleteAllMenu.removeAttribute('disabled');
  1122.   }
  1123.   else
  1124.   {
  1125.     detachAllMenu.setAttribute('disabled', 'true');
  1126.     deleteAllMenu.setAttribute('disabled', 'true');
  1127.   }
  1128. }
  1129.  
  1130. // Public method used to clear the file attachment menu
  1131. function ClearAttachmentMenu(popup) 
  1132.   if ( popup ) 
  1133.   { 
  1134.      while ( popup.childNodes[0].localName == 'menu' )
  1135.        popup.removeChild(popup.childNodes[0]); 
  1136.   } 
  1137. }
  1138.  
  1139. // Public method used to determine the number of attachments for the currently displayed message...
  1140. function GetNumberOfAttachmentsForDisplayedMessage()
  1141. {
  1142.   return currentAttachments.length;
  1143. }
  1144.  
  1145. // private method used to build up a menu list of attachments
  1146. function addAttachmentToPopup(popup, attachment, attachmentIndex) 
  1147.   if (popup)
  1148.   { 
  1149.     var item = document.createElement('menu');
  1150.     if ( item ) 
  1151.     {     
  1152.       if (!gMessengerBundle)
  1153.         gMessengerBundle = document.getElementById("bundle_messenger");
  1154.  
  1155.       // insert the item just before the separator
  1156.       item = popup.insertBefore(item, popup.childNodes[attachmentIndex - 1]);
  1157.       item.setAttribute('class', 'menu-iconic attachment-item');
  1158.  
  1159.       var formattedDisplayNameString = gMessengerBundle.getFormattedString("attachmentDisplayNameFormat",
  1160.                                        [attachmentIndex, attachment.displayName]);
  1161.  
  1162.       item.setAttribute('label', formattedDisplayNameString); 
  1163.       item.setAttribute('accesskey', attachmentIndex); 
  1164.       if (attachment.contentType == "text/x-moz-deleted") {
  1165.         item.setAttribute('disabled', 'true');
  1166.         return;
  1167.       }
  1168.       setApplicationIconForAttachment(attachment, item);
  1169.       var canDetach = CanDetachAttachments();
  1170.  
  1171.       var openpopup = document.createElement('menupopup');
  1172.       openpopup = item.appendChild(openpopup);
  1173.  
  1174.       var menuitementry = document.createElement('menuitem');     
  1175.  
  1176.       menuitementry.attachment = attachment;
  1177.       menuitementry.setAttribute('oncommand', 'this.attachment.openAttachment()'); 
  1178.  
  1179.       if (!gDeleteLabel)
  1180.         gDeleteLabel = gMessengerBundle.getString("deleteLabel");
  1181.       if (!gDeleteLabelAccesskey)
  1182.         gDeleteLabelAccesskey = gMessengerBundle.getString("deleteLabelAccesskey");
  1183.       if (!gDetachLabel)
  1184.         gDetachLabel = gMessengerBundle.getString("detachLabel");
  1185.       if (!gDetachLabelAccesskey)
  1186.         gDetachLabelAccesskey = gMessengerBundle.getString("detachLabelAccesskey");
  1187.       if (!gSaveLabel)
  1188.         gSaveLabel = gMessengerBundle.getString("saveLabel");
  1189.       if (!gSaveLabelAccesskey)
  1190.         gSaveLabelAccesskey = gMessengerBundle.getString("saveLabelAccesskey");
  1191.       if (!gOpenLabel)
  1192.         gOpenLabel = gMessengerBundle.getString("openLabel");
  1193.       if (!gOpenLabelAccesskey)
  1194.         gOpenLabelAccesskey = gMessengerBundle.getString("openLabelAccesskey");
  1195.  
  1196.       menuitementry.setAttribute('label', gOpenLabel); 
  1197.       menuitementry.setAttribute('accesskey', gOpenLabelAccesskey); 
  1198.       menuitementry = openpopup.appendChild(menuitementry);
  1199.  
  1200.       menuitementry = document.createElement('menuitem');
  1201.       menuitementry.attachment = attachment;
  1202.       menuitementry.setAttribute('oncommand', 'this.attachment.saveAttachment()'); 
  1203.       menuitementry.setAttribute('label', gSaveLabel); 
  1204.       menuitementry.setAttribute('accesskey', gSaveLabelAccesskey); 
  1205.       menuitementry = openpopup.appendChild(menuitementry);
  1206.  
  1207.       var menuseparator = document.createElement('menuseparator');
  1208.       openpopup.appendChild(menuseparator);
  1209.  
  1210.       menuitementry = document.createElement('menuitem');
  1211.       menuitementry.attachment = attachment;
  1212.       menuitementry.setAttribute('oncommand', 'this.attachment.detachAttachment()'); 
  1213.       menuitementry.setAttribute('label', gDetachLabel); 
  1214.       menuitementry.setAttribute('accesskey', gDetachLabelAccesskey); 
  1215.       if (!canDetach)
  1216.         menuitementry.setAttribute('disabled', 'true');
  1217.       menuitementry = openpopup.appendChild(menuitementry);
  1218.  
  1219.       menuitementry = document.createElement('menuitem');
  1220.       menuitementry.attachment = attachment;
  1221.       menuitementry.setAttribute('oncommand', 'this.attachment.deleteAttachment()'); 
  1222.       menuitementry.setAttribute('label', gDeleteLabel); 
  1223.       menuitementry.setAttribute('accesskey', gDeleteLabelAccesskey); 
  1224.       if (!canDetach)
  1225.         menuitementry.setAttribute('disabled', 'true');
  1226.       menuitementry = openpopup.appendChild(menuitementry);
  1227.     }  // if we created a menu item for this attachment...
  1228.   } // if we have a popup
  1229.  
  1230. function HandleMultipleAttachments(commandPrefix, selectedAttachments)
  1231. {
  1232.  try 
  1233.  {
  1234.    // convert our attachment data into some c++ friendly structs
  1235.    var attachmentContentTypeArray = new Array();
  1236.    var attachmentUrlArray = new Array();
  1237.    var attachmentDisplayNameArray = new Array();
  1238.    var attachmentMessageUriArray = new Array();
  1239.  
  1240.    // populate these arrays..
  1241.    for (index in selectedAttachments)
  1242.    {
  1243.      var attachment = selectedAttachments[index].attachment;
  1244.      attachmentContentTypeArray[index] = attachment.contentType;
  1245.      attachmentUrlArray[index] = attachment.url;
  1246.      attachmentDisplayNameArray[index] = encodeURI(attachment.displayName);
  1247.      attachmentMessageUriArray[index] = attachment.uri;
  1248.    }
  1249.  
  1250.    // okay the list has been built...now call our action code...
  1251.    switch (commandPrefix)
  1252.    {
  1253.      case "saveAttachment":
  1254.        messenger.saveAllAttachments(attachmentContentTypeArray.length,
  1255.                                     attachmentContentTypeArray,
  1256.                                     attachmentUrlArray,
  1257.                                     attachmentDisplayNameArray,
  1258.                                     attachmentMessageUriArray);
  1259.        break;
  1260.      case "detachAttachment":
  1261.        messenger.detachAllAttachments(attachmentContentTypeArray.length,
  1262.                                       attachmentContentTypeArray,
  1263.                                       attachmentUrlArray,
  1264.                                       attachmentDisplayNameArray,
  1265.                                       attachmentMessageUriArray,
  1266.                                       true /* save */);
  1267.        break;
  1268.      case "deleteAttachment":
  1269.        messenger.detachAllAttachments(attachmentContentTypeArray.length,
  1270.                                       attachmentContentTypeArray,
  1271.                                       attachmentUrlArray,
  1272.                                       attachmentDisplayNameArray,
  1273.                                       attachmentMessageUriArray,
  1274.                                       false /* don't save */);
  1275.        break;
  1276.      default:
  1277.        dump (commandPrefix + "** unknown handle all attachments action **\n");
  1278.    }
  1279.  }
  1280.  catch (ex)
  1281.  {
  1282.    dump ("** failed to handle all attachments **\n");
  1283.  }
  1284. }
  1285.  
  1286. function ClearAttachmentList() 
  1287.   // we also have to disable the File/Attachments menuitem
  1288.   node = document.getElementById("fileAttachmentMenu");
  1289.   if (node)
  1290.     node.setAttribute("disabled", "true");
  1291.  
  1292.   // clear selection
  1293.   var list = document.getElementById('attachmentList');
  1294.   list.clearSelection();
  1295.  
  1296.   while (list.hasChildNodes()) 
  1297.     list.removeChild(list.lastChild);
  1298. }
  1299.  
  1300. function ShowEditMessageButton() 
  1301. {
  1302.   var editBox = document.getElementById("editMessageBox");
  1303.   if (editBox)
  1304.     editBox.collapsed = false;
  1305.  
  1306. function ClearEditMessageButton() 
  1307.   var editBox = document.getElementById("editMessageBox");
  1308.   if (editBox)
  1309.     editBox.collapsed = true;
  1310. }
  1311.  
  1312.  
  1313. var attachmentAreaDNDObserver = {
  1314.   onDragStart: function (aEvent, aAttachmentData, aDragAction)
  1315.   {
  1316.     var target = aEvent.target;
  1317.     if (target.localName == "listitem")
  1318.     {
  1319.       var attachmentUrl = target.getAttribute("attachmentUrl");
  1320.       var attachmentDisplayName = target.getAttribute("label");
  1321.       var attachmentContentType = target.getAttribute("attachmentContentType");
  1322.       var tmpurl = attachmentUrl;
  1323.       var tmpurlWithExtraInfo = tmpurl + "&type=" + attachmentContentType + "&filename=" + attachmentDisplayName;
  1324.       aAttachmentData.data = new TransferData();
  1325.       if (attachmentUrl && attachmentDisplayName)
  1326.       {
  1327.         aAttachmentData.data.addDataForFlavour("text/x-moz-url", tmpurlWithExtraInfo + "\n" + attachmentDisplayName);
  1328.         aAttachmentData.data.addDataForFlavour("text/x-moz-url-data", tmpurl);
  1329.         aAttachmentData.data.addDataForFlavour("text/x-moz-url-desc", attachmentDisplayName);
  1330.         
  1331.         aAttachmentData.data.addDataForFlavour("application/x-moz-file-promise-url", tmpurl);   
  1332.         aAttachmentData.data.addDataForFlavour("application/x-moz-file-promise", new nsFlavorDataProvider(), 0, Components.interfaces.nsISupports);     
  1333.       }
  1334.     }
  1335.   }
  1336. };
  1337.  
  1338. function nsFlavorDataProvider()
  1339. {
  1340. }
  1341.  
  1342. nsFlavorDataProvider.prototype =
  1343. {
  1344.   QueryInterface : function(iid)
  1345.   {
  1346.       if (iid.equals(Components.interfaces.nsIFlavorDataProvider) ||
  1347.           iid.equals(Components.interfaces.nsISupports))
  1348.         return this;
  1349.       throw Components.results.NS_NOINTERFACE;
  1350.   },
  1351.   
  1352.   getFlavorData : function(aTransferable, aFlavor, aData, aDataLen)
  1353.   {
  1354.  
  1355.     // get the url for the attachment
  1356.     if (aFlavor == "application/x-moz-file-promise")
  1357.     {
  1358.       var urlPrimitive = { };
  1359.       var dataSize = { };
  1360.       aTransferable.getTransferData("application/x-moz-file-promise-url", urlPrimitive, dataSize);
  1361.  
  1362.       var srcUrlPrimitive = urlPrimitive.value.QueryInterface(Components.interfaces.nsISupportsString);
  1363.  
  1364.       // now get the destination file location from kFilePromiseDirectoryMime
  1365.       var dirPrimitive = {};
  1366.       aTransferable.getTransferData("application/x-moz-file-promise-dir", dirPrimitive, dataSize);
  1367.       var destDirectory = dirPrimitive.value.QueryInterface(Components.interfaces.nsILocalFile);
  1368.  
  1369.       // now save the attachment to the specified location
  1370.       // XXX: we need more information than just the attachment url to save it, fortunately, we have an array
  1371.       // of all the current attachments so we can cheat and scan through them
  1372.  
  1373.       var attachment = null;
  1374.       for (index in currentAttachments)
  1375.       {
  1376.         attachment = currentAttachments[index];
  1377.         if (attachment.url == srcUrlPrimitive)
  1378.           break;
  1379.       }
  1380.  
  1381.       // call our code for saving attachments
  1382.       if (attachment)
  1383.       {
  1384.         var destFilePath = messenger.saveAttachmentToFolder(attachment.contentType, attachment.url, encodeURIComponent(attachment.displayName), attachment.uri, destDirectory);
  1385.         aData.value = destFilePath.QueryInterface(Components.interfaces.nsISupports);
  1386.         aDataLen.value = 4;
  1387.       }
  1388.     }
  1389.   }
  1390.  
  1391. }
  1392.  
  1393. function nsDummyMsgHeader()
  1394. {
  1395. }
  1396.  
  1397. nsDummyMsgHeader.prototype =
  1398. {
  1399.   mProperties : new Array,
  1400.   getStringProperty : function(property) {return this.mProperties[property];},
  1401.   setStringProperty : function(property, val) {this.mProperties[property] = val;},
  1402.   messageSize : 0,
  1403.   recipients : null,
  1404.   from : null,
  1405.   subject : null,
  1406.   ccList : null,
  1407.   messageId : null,
  1408.   accountKey : "",
  1409.   folder : null
  1410. };
  1411.